Go语言从Docker源码学习Go function和method
method是针对某一类型定义的function,
function可以单独调用,method必须针对某一类型的实例进行调用
//function 调用方式 packageName.FuncName() //method 调用方式 var t packageName.Type t.MethodName() 源码 func (cli *DockerCli) LoadConfigFile() (err error) { cli.configFile, err = registry.LoadConfig(os.Getenv("HOME")) if err != nil { fmt.Fprintf(cli.err, "WARNING: %s\n", err) } return err } func NewDockerCli(in io.ReadCloser, out, err io.Writer, proto, addr string, tlsConfig *tls.Config) *DockerCli { ... return &DockerCli{ proto: proto, addr: addr, in: in, out: out, ... } }LoadConfigFile()是method,而NewDockerCli()是function。
Function func functionName(parameter_list) (return_value_list) { … } //parameter_list form (param1 type1, param2 type2, …) //return_value_list form (ret1 type1, ret2 type2, …)注:function可以返回多个参数,如果在调用时不想接受某个参数,使用下划线_
_, exist = FunctionName(...) 值/引用传递的问题默认情况下,function中的参数是进行值传递的,如果想要在function内部修改原来数据的值,需要传递引用,用&符号。
例外:引用类型默认是进行引用传递的,比如slices, maps, interfaces, channels.
func ChangeValue(value *Struct1, newValue int) { value.Int1 = newValue } func NotChangeValue(value Struct1, newValue int) { value2 := &value value2.Int1 = newValue }可以通过定义最后一个参数的类型为...Type来传递多个同一类型的多个参数
func Min(ints ...int) int {//ints类型为[]int{},值是传递进来的值。 if len(ints) == 0 { return -1 } min := ints[0] for _, x := range ints { if x < min { min = x } } return min } //调用 fmt.Println(Min(4, 6, 9, 10, 3, 2)) arr := []int{9, 5, 3, 10, 20, 2} fmt.Println(Min(arr...))
上面介绍的是传入多个相同类型的情况,那如果不同类型的多个怎么定义呢?
有两种方法,第一种是通过定义一个新的struct,包含这些不同的类型,然后把这个struct类型作为输入参数;第二种是使用空的Interface。
主要介绍第二种
func PrintType(variables ...interface{}) { for _, v := range variables { switch v.(type) { case int: fmt.Println("type is int %d", v) default: fmt.Println("other type %v", v) } } } func showFunctionMultiInterfaceParameters() { lemon.PrintType(5, "aaaa") var2 := []interface{}{6, 7, 9, "bbb", "ccc"} lemon.PrintType(var2...) }注:需要在方法内部判断类型,之后进行操作。
deferdefer用来定义在方法最后执行的语句,在return后,}之前。
注意:在同一个方法中先后定义两个defer,执行顺序是倒序的,LIFO.
func DeferOrder() { for i := 0; i < 4; i++ { defer fmt.Println("index value is %d", i) } } //输出 index value is %d 3 index value is %d 2 index value is %d 1 index value is %d 0 methodmethod是特殊的function,定义在某一特定的类型上,通过类型的实例来进行调用,这个实例被叫receiver。
receiver类型可以是任意类型,包括function类型,但是,receiver不能是interface类型。
func (recv receiver_type) methodName(parameter_list) (return_value_list) {...}注:method和它的类型必须定义在一个包内!但是我们可以通过其它的方式来实现。
解决思路:怎么样可以让原来的类型在我们自己的包中定义呢?
两种方法:
第一种,通过alias。
type Int int这样在对Int进行方法定义。
第二种,通过匿名属性
type Int struct{ int }两种方法,第二种比较好,因为第一种只针对当前的alias有用,重用性不如第二种方法好。
如果想要在method内部修改receiver的属性值,recv *receiver_type(在type前面增加一个*),表示传入引用。
此时传入值也没关系,go语言自动实现转化,t.Func转换为(&t).Func.
func (b *B) change(){...} var b1 B b1.change() 总结go语言中把method和type分开进行定义。
好处是我们可以为任何类型增加方法,而不用去修改原来的代码。
相关热词: Go语言
本站内容来源于网络,如有侵权请与我们联系,我们会及时删除,我们深感抱歉!
注:本站所有信息仅供用于网络技术学习参考,学习中请遵循相关法律法规!
本文地址: https://v30.fanwenzhu.com/jiaob/go/9836.shtml
相关文章
热门TAG
win10 ecshop 主机 阿里云 解决 配置 C# C++ 解析 SQL语句 命令 Go语言 方法 CSS3 HTML5 CSS win7 MSSQL 服务器配置 IIS7.5 IIS7 IIS6 IIS CentOS 7 Linux oracle数据库 oracle phpcms discuz discuz教程最新文章
-
Lisp进修Windows下面的开拓情
时间:2021-01-12
-
Lisp进修Windows下面的开拓情
时间:2021-01-12
-
为Go语言GC正名-2秒到1毫
时间:2020-12-27
-
go语言初探 一个helloworld编
时间:2020-12-27
热门文章
-
为Go语言GC正名-2秒到1毫秒的演变史
时间:2020-12-27
-
Lisp进修Windows下面的开拓情况搭建
时间:2021-01-12
-
Lisp进修Windows下面的开拓情况搭建
时间:2021-01-12
-
go语言初探 一个helloworld编译出来有2.2M!
时间:2020-12-27
